手把手教你做個人 app

我們都知道,開發一個app很大程度依賴服務端:服務端提供接口數據,然後我們展示;另外,開發一個app,還需要美工協助切圖。沒了接口,沒了美工,app似乎只能做成單機版或工具類app,真的是這樣的嗎?先來展示下我的個人app,沒有服務端,沒有美工完成的,換言之,我幹了所有人的活:

手把手教你做個人 app

這個app叫“微言”,他對於我意義很重大,最初微言只是我一個練手的項目,剛剛工作,技術有限,微言只是sqlite記事本類app,只能本地操作,後來慢慢演變現在幾近完美的app,從中我學到很多,熟悉了立項到上線的整個流程,最新技術得以實踐,從一個程序猿思維從而向產品思維轉變,簡單的Photoshop等。當然長期的積累自然會帶來經濟方面的收益,這裡秀下我的app廣告收益,我的所有app之和:

手把手教你做個人 app

最多一個月4000多,4000多什麼概念,比我當時薪資都高呢,這些“成就”有了我可以在此吹牛的資本,哈哈哈!

接下來,我一一分析,帶您完成這樣的一個完整的app。

沒有服務端

jsoup

我無意聽到大牛同事說到解析html,比較有興趣去搜索這是什麼玩意兒,知道了一個強大的東西jsoup,jsoup能解析html,即網站,於是我的微言脫離了單機版。對用戶而言,他不在乎數據從何而來,管您是從接口取的還是解析html,他們關心的是app體驗和功能的完善。我就這樣瞞天過海,數據取之網頁了,群裡之前太多人太多人問我用的什麼服務器,回覆太多次解析html後就不願意再回復了。

我選擇這種方式有個最大的好處就是數據不需要本人維護,巧妙地避開了我不會服務端開發,更不需要做接口;解析html也有個最大的弊端,一旦對方網站節點變化了,或許您的app就掛了,必須及時去更新。

使用方法

步驟一

首先網絡請求,這裡用的Retrofit,具體見:Android MVP+Retrofit+RxJava實踐小結。以解析我的博客http://wuxiaolong.me/ 示例,可以拿到類似以下數據:

手把手教你做個人 app

在谷歌瀏覽器,我的博客頁面點擊右鍵-查看網頁源代碼(V),同樣看到以上數據。

步驟二

1、app/build.gradle

compile 'org.jsoup:jsoup:1.10.1'

2、解析html

要訣:多觀察html節點、標籤。

先觀察我們要解析的數據(以我的博客http://wuxiaolong.me/ 示例),首頁分別有標題、發表時間、文章分類、文章評論、文章摘要5個元素谷歌瀏覽器,我們這次只需要標題、發表時間、文章摘要;可以看到我的博客是分頁,第一頁網址是http://wuxiaolong.me 和第二頁網址卻是http://wuxiaolong.me/page/2/ ,之後區別就是頁碼,因此app做分頁的話要判斷第一頁和其他頁,最終我做成的效果:

手把手教你做個人 app

我們一一分析,關於jsoup語法,這裡不說,見 jsoup官網。

(1)標題數據結構如下:

觀察可根據class="post-title"的getElementsByClass解析:

//responseBody是retrofit網絡請求返回的,轉成String,即我們需要解析的數據Document document = Jsoup.parse(new String(responseBody.bytes(), "UTF-8"));List<element> titleElementList = new ArrayList<>();Elements titleElements = document.getElementsByClass("post-title-link");for (Element element : titleElements) {
titleElementList.add(element);
//text拿到文本,如這裡的“Android App Shortcuts”
LogUtil.d("text=" + element.text());
//拿到href屬性值,如這裡“/2016/10/31/AppShortcuts/”,即博客鏈接,如果跳轉詳情需要加上“http://wuxiaolong.me”
LogUtil.d("href=" + element.attr("href"));}/<element>

(2)發表時間數據結構如下:

發表於<time>2016-10-31/<time>

觀察,同樣用getElementsByClass解析:

List<element> timeElementList = new ArrayList<>();Elements timeElements = document.getElementsByClass("post-time");for (Element element : timeElements) {
//這裡通過解析"time"標籤,然後取文本,即“2016-10-31”
LogUtil.d("time=" + element.getElementsByTag("time").text());
timeElementList.add(element);}/<element>

(3)文章摘要數據結構如下:

簡介

Android 7.1允許您定義應用程序中特定操作的快捷方式。這些快捷鍵可以顯示桌面,例如Nexus和Pixel設備。快捷鍵可讓您的用戶在應用程序中快速啟動常見或推薦的任務。
每個快捷鍵引用一個或多個意圖,每個意圖在用戶選擇快捷方式時在應用程序中啟動特定操作。可以表達為快捷方式的操作示例包括:

恩,也是用的getElementsByClass解析:

List<element> bodyElementList = new ArrayList<>();Elements bodyElements = document.getElementsByClass("post-body");for (Element element : bodyElements) {
//這裡用text()只是拿到文本,但我想要直接返回html標籤,很好,jsoup有html()方法。
LogUtil.d("body=" + element.html());
bodyElementList.add(element);}/<element>

3、核心代碼

private void loadMyBlog() {
Call<responsebody> call;
//分頁處理
if (page == 1) {
call = apiStores.loadMyBlog();
} else {
call = apiStores.loadMyBlog(page);
}
call.enqueue(new RetrofitCallback<responsebody>() {
@Override
public void onSuccess(ResponseBody responseBody) {
try {
Document document = Jsoup.parse(new String(responseBody.bytes(), "UTF-8"));
List<element> titleElementList = new ArrayList<>();
Elements titleElements = document.getElementsByClass("post-title-link");
for (Element element : titleElements) {
titleElementList.add(element);

//LogUtil.d("text=" + element.text());
//LogUtil.d("href=" + element.attr("href"));
}
List<element> timeElementList = new ArrayList<>();
Elements timeElements = document.getElementsByClass("post-time");
for (Element element : timeElements) {
//LogUtil.d("time=" + element.getElementsByTag("time").text());
timeElementList.add(element);
}
//Elements categoryElements = document.getElementsByClass("post-category");
//for (Element element : categoryElements) {
// LogUtil.d("category=" + element.getElementsByTag("a").text());
//}
List<element> bodyElementList = new ArrayList<>();
Elements bodyElements = document.getElementsByClass("post-body");
for (Element element : bodyElements) {
LogUtil.d("body=" + element.html());
bodyElementList.add(element);
}
if (page == 1) {
dataAdapter.clear();
}
dataAdapter.addAll(titleElementList, timeElementList, bodyElementList);
if (titleElementList.size() < 8) {
//因為我的博客一頁8條數據,當小於8時,說明沒有下一頁了
pullLoadMoreRecyclerView.setHasMore(false);
} else {
pullLoadMoreRecyclerView.setHasMore(true);
}
} catch (Exception e) {
e.printStackTrace();
}
}
@Override
public void onFailure(int code, String msg) {
toastShow(msg);
}
@Override
public void onThrowable(Throwable t) {
toastShow(t.getMessage());
}
@Override
public void onFinish() {
pullLoadMoreRecyclerView.setPullLoadMoreCompleted();
}
});
addCalls(call);}/<element>/<element>/<element>/<responsebody>/<responsebody>

jsoup解析源碼

解析我的博客源碼已經上傳我的github,見:https://github.com/WuXiaolong/WeWin

想必這樣一一分析,您一定會jsoup解析html,如果還不會,私下給我發個大紅包,我再手把手教您,發個超大紅包,今晚我就是您的啦,嘿嘿。

題外

可能您擔心,jsoup解析html,這樣爬蟲難道不侵權嗎?是的,我也擔心,所以我的app也只在我的群裡“宣傳宣傳”。

bmob

仔細的您,肯定發現了,jsoup爬數據,只能做展示功能,那我的微言裡不是有評論功能嘛!這是怎麼做到的呢?我剛開始做app那會,個人app是做不了POST功能的,但bmob出現解決了個人開發者這個沒有服務器的痛點,它就相當於一個數據庫,提供了sdk,您可以做增刪改查操作,我們只需要專注客戶端,後端就不用管了,話說如此,我們還是稍微懂點數據庫知識的,以便於我們更好利用bmob。除了bmob,現在還有leancloud、apicloud等都有類似的服務,很感謝他們,及時解決我們的剛需,個人app還可以有一線生機。

關於bmob、leancloud、apicloud如何使用,我知道聰明的您已經在看他們的官方文檔了。

沒有美工

美工切圖

在實際開發中,有些效果,只需要美工做張圖片就能輕鬆搞定,沒有美工切圖的配合,app開發似乎難以進展下去了,是嗎?其實我在《Android Design Support Library使用》一文提到一句話:“目前這個sample,Material design風格的效果都有了,相當一個空殼子,您只需在實際開發中塞真實數據就是一個perfect app”,對,就用谷歌的Material design風格就OK了,它提供了一套規範、圖片,足夠我們個人app使用了,而且現在還有vector,更是強大之極。比如微言-每日推薦右上角的刷新按鈕,如圖:

手把手教你做個人 app

相應的xml:

<menu>xmlns:app="http://schemas.android.com/apk/res-auto">
<item>android:id="@+id/action_refresh"
android:icon="@drawable/ic_loop_24dp"
android:title="刷新"
app:showAsAction="always" />/<item>/<menu>

平時ic_loop_24dp肯定是張圖片,然而我用的vector,代碼如下:

<vector>android:width="24dp"
android:height="24dp"
android:viewportWidth="24.0"
android:viewportHeight="24.0">
<path>android:fillColor="?attr/colorControlNormal"
android:pathData="M12,4L12,1L8,5l4,4L12,6c3.31,0 6,2.69 6,6 0,1.01 -0.25,1.97 -0.7,2.8l1.46,1.46C19.54,15.03 20,13.57 20,12c0,-4.42 -3.58,-8 -8,-8zM12,18c-3.31,0 -6,-2.69 -6,-6 0,-1.01 0.25,-1.97 0.7,-2.8L5.24,7.74C4.46,8.97 4,10.43 4,12c0,4.42 3.58,8 8,8v3l4,-4 -4,-4v3z"/>/<path>/<vector>

是不是很神奇,vector如何創建:

手把手教你做個人 app

這裡隨便演示,創建了一個vector,關於vector學習可參考醫生的《Android Vector曲折的兼容之路》一文,寫的很詳細,這裡不多說了。

app圖標

app當然希望有個漂亮的有意義的圖標,會用到Photoshop,我當然不會啊,那必須得學啊。一般安卓市場需要圖標尺寸16*16,48*48,512*512,圓角,Android開發需要48*48,72*72,96*96,144*144,196*196,因此PS的時候,只需要做個最大尺寸512*512,然後縮小即可。

圖標的PS步驟:

1、用photoshop打開您想修改的圖片

2、在左側工具欄中選擇“圓角矩形工具”(默認的是“矩形工具”,您只需要右擊圖標就可以發現“圓角矩形工具”),如上圖

3、在上面“半徑”框中輸入您想要的圓角半徑,一般圖片選25即可,為了效果明顯我設置為40,看上圖有一個框顯示半徑40.

4、在打開的圖片上畫一個圓角矩形,把圖片覆蓋住。

5、對著已經被覆蓋的圖片選區右擊,選擇“建立選區”,如果有窗口彈出直接點擊“確定”,在彈出的選項中直接點擊“確認”

6、在上方的“選擇”選項卡中點擊,在下拉框中找到“反向”,也可以使用快捷鍵ctrl+shilf+i。

7、在右下方的圖層欄中雙擊“背景”圖片(上面第一張圖片右下角可以看到),如果有窗口彈出直接點擊“確定”,完成解鎖。

8、按鍵盤上的"DELETE"鍵清除四個直角。

9、繼續右擊“形狀1”(在畫面右下方圖層那裡可以找到),在彈出選項中選擇“刪除圖層”,如果有窗口彈出直接點擊“是”。

10、OK,您可以看到一個圓角圖片。

11、最後在左上角點擊文件--》存儲為--》選擇png格式(其他格式也可以),完成。

手把手教你做個人 app

為什麼微言的圖標是一個“言”字,因為我覺得這樣簡潔大方,簡單明瞭,言簡意賅……算了,不裝了,其他我不會P啊!

推廣技巧

做完一個個人app,是不是很有成就感啊,上線安卓市場,卻沒幾個下載量,尼瑪,勞資花了很長時間呢,這麼牛B的app怎麼沒人下載?!心情一下子跌倒谷底,那得讓更多的人知道自己的app啊,我是這樣做的:

1、邀請好評

您去下載一個app時,可能會看下這個app的評論,如果有很多好評,您會不會更願意去下載呢,是的,來看我的微言好評如潮:

手把手教你做個人 app

哈哈哈,是不是很牛,邀請評論我寫成了工具類了,請笑納:

public class InviteCommentUtil {
private String mDateFormat = "yyyy-MM-dd";
private String mInviteCommentTime;
/** * 選擇哪天彈邀請評論框 */
public void startComment(final Activity activity) {
mInviteCommentTime = PreferenceUtils.getPreferenceString(activity, "inviteCommentTime", TimeUtil.getCurrentTime(mDateFormat));
String saveCommentTime = PreferenceUtils.getPreferenceString(activity, "saveCommentTime", TimeUtil.getCurrentTime(mDateFormat));
int compareDateValue = TimeUtil.compareDate(mInviteCommentTime, saveCommentTime, mDateFormat);
if (compareDateValue == 1) {
AlertDialog.Builder builder = new AlertDialog.Builder(
activity);
int nowReadingTotal = ReadUtil.getReadedTotal();
builder.setMessage(Html.fromHtml("您已經累計閱讀" + nowReadingTotal + "/<font>字,再接再厲哦!如果喜歡我,希望您能在應用市場給予五星/<font>好評!"));
builder.setTitle("求好評");
builder.setPositiveButton("好評鼓勵",
new android.content.DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
setComment(activity);
try {
Intent intent = new Intent(
"android.intent.action.VIEW");
intent.setData(Uri
.parse("market://details?id=com.android.xiaomolongstudio.weiyan"));
activity.startActivity(intent);
activity.overridePendingTransition(
R.anim.enter_right_to_left, R.anim.exit);
} catch (Exception e) {
e.printStackTrace();
}
dialog.dismiss();
}
});
builder.setNegativeButton("下次再說",
new android.content.DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
setComment(activity);
dialog.dismiss();

}
});
builder.create().show();
}
}
/** * 保存,直到下週再提示 */
private void setComment(Activity activity) {
PreferenceUtils.setPreferenceString(activity, "saveCommentTime", mInviteCommentTime);
}}

注意:這裡邀請評論時間控制好,不能太頻繁了,不然用戶會反感的。

2、專注一個市場

不知道您有沒有發現,某個市場您明明沒有發佈,卻能搜到您的app,沒錯,一些市場會抓您的這個app,比如豌豆莢,百度,因此策略,專注一個市場,這個市場出名了,還怕其他市場不知道嗎?當然我們是爭取每個市場都能發佈上線,多一個下載是一個。

微言位於分類下前排

手把手教你做個人 app

多次進入精品系列

手把手教你做個人 app

如何進入前排或精品,邀請好評是關鍵的一步。

3、新品推薦

新品上線,很多市場有新品推薦,比如小米、妹紙、360、應用寶,一般需要自薦,一旦被推薦,下載量是可觀的,微言肯定被推薦過啦。哦,更新版本也算新品哦。

4、微博

這是我好哥們教我的,他真的好牛,個人app做的更牛,給您們看一個鏈接:http://weibo.com/p/1008082a702d2cb6146485e5dc3d624d31def6 ,就知道如何在微博上推廣了,沒錯,就是話題,用兩個#號圈起來,發微博,就是一個話題,別人可以這個話題下討論,無形中形成了推廣作用。

5、app分享

app分享功能肯定是要做的,萬一用戶覺得您的這個app很棒,想推薦給好友,結果沒分享功能,豈不歇菜,分享微博可以加兩個#號圈起來哦。

6、QQ群

如果您直接群裡發app的下載鏈接,只會一個結果:被T。像我們程序猿,大部分都是技術群,我的做法是寫文章分享我app用的技術,文章會附上app下載地址,然後有這個技術興趣的人可能詢問,這樣我就名正言順地“推廣”了,哈哈,我好壞!

以上僅我知道的,不一定有效,畢竟我不是專業的推廣人員。

如何賺錢

萬事俱備,只欠東風,當您的用戶量足夠大的時候,自然有人找您,投資您的app,這個過程前期0收益,耗得精力時間不算,可能還得燒錢,不適合個人開發者,我選擇的是賺賺小錢,app加廣告的方式。

廣告平臺選擇第三方的,最早我使用的多盟,但是一段時間發現我根本沒什麼收益,感覺多盟扣量好嚴重,之後嘗試多易傳媒、艾德思奇、芒果聚合,也使用過點樂積分,發現一些市場不讓上線積分類app,放棄之,還有百度移動廣告聯盟、騰訊的廣點通,谷歌的也玩過,收益最穩定屬百度的,這點不黑它。

至於廣告集成,也是提供sdk,自行去他們的官網瞭解吧。

最後

我的幾個app,都是我個人做到了很滿意,功能都很完善,以至於我後來覺得沒什麼好更新的,再加上廣告收益好也就一段時間,移動廣告商雨後春筍,導致廣告單價太低,而且安卓市場對個人開發者各種限制,比如不讓上線視頻類app;某度,某60必須用自家的加固才讓上線app等,就沒什麼動力繼續維護app,做事還是要有動力的,不然活著幹嗎?不過我知道有人還在堅持做個人app,做的好的,日入幾百甚至上千的。app最終目的就是賺錢。


分享到:


相關文章: