1.需求分析


(1)业务需求分析:分析潜在客户需要什么产品或服务
(2)架构分析:分析所做的产品或服务需要什么功能及实现方式并画出功能结构图,记事本案例如下

(3)数据库类设计分析:数据库设计是项目开发中非常关键的一个环节。同样在记事本案例中也至关重要,我们通过数据库表(Note)进行增删改查操作,记事本的数据表如下所示

字段名 数据类型 字段名 数据类型
id integer 编号
content text 事件内容
notetime text 保存事件的时间

(4)界面需求分析:友好的界面在移动平台开发中非常重要,也是用户使用软件的先决条件。记事本案例分为3个界面,分别为记事本界面、添加界面和修改界面

记事本界面包含添加按钮和记录列表,点击后会跳转到添加界面,界面标题为添加记录,也可在该界面清除和保存编辑的内容。当点击记事本界面中的Item时,会跳转到修改记录界面,界面标题为修改记录,在该界面中可以查看和修改已保存的记录内容,也可清除和保存编辑的内容

2.记事本案例步骤

(1)创建项目后将Activity名称设置为NotepadActivity,布局文件为activity_notepad并将所需素材add.png(添加按钮),save_note.png(保存按钮),delete.png(删除按钮),back.png(返回按钮)导入drawable文件夹中,如下

上面添加按钮右边还有个返回按钮图片是白色的

(2)编写activity_notepad.xml布局文件,放置一个TextView控件用于显示界面标题,一个ListView控件用于显示记录列表,一个ImageView控件用于显示添加按钮的图片

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:background="#fefefe"tools:context=".NotepadActivity"><TextViewandroid:layout_width="match_parent"android:layout_height="45dp"android:gravity="center"android:text="记事本"android:textSize="20sp"android:textColor="@android:color/white"android:textStyle="bold"android:background="#fb7a6a"android:id="@+id/note_name"/><ListViewandroid:layout_width="match_parent"android:layout_height="match_parent"android:id="@+id/listview"android:cacheColorHint="#00000000"android:divider="#B4B4B4"android:dividerHeight="1dp"android:fadingEdge="none"android:listSelector="#00000000"android:scrollbars="none"android:layout_below="@id/note_name"/><ImageViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:src="@drawable/add"android:id="@+id/add"android:layout_marginBottom="30dp"android:layout_alignParentBottom="true"android:layout_centerHorizontal="true"/></RelativeLayout>

(3)修改清单文件
项目创建后所有界面都有一个默认的标题栏,该标题栏不太美观,因此需要在清单文件(AndroidManifest.xml)中的<application>标签中修改android:theme属性

android:theme="@style/Theme.AppCompat.NoActionBar"

(4)搭建记事本界面Item布局
在res/layout文件夹中,创建一个布局文件notepad_item_layout.xml,在其中放置两个TextView控件,分别用来显示记录的部分内容与保存记录的时间

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"android:paddingLeft="12dp"><TextViewandroid:layout_width="match_parent"android:layout_height="wrap_content"android:id="@+id/item_content"android:maxLines="2"android:ellipsize="end"android:lineSpacingExtra="3dp"android:paddingTop="10dp"android:textColor="@android:color/black"/><TextViewandroid:layout_width="match_parent"android:layout_height="wrap_content"android:id="@+id/item_time"android:textColor="#fb7a6a"android:paddingTop="5dp"android:paddingBottom="7dp"/>
</LinearLayout>

(5)封装记录信息实体类
由于记事本中的每个记录都会有记录内容和保存记录的时间属性,因此需要创建一个NotepadBean用于存放这些属性。选中你所创建的项目的包,右击选择New>Package,创建一个bean包,在该包中创建一个NotepadBean类,该类中定义记录信息的所有属性

package com.example.notepad.bean;public class NotepadBean {private String id;private String notepadContent;private String notepadTime;public String getId(){return id;}public void setId(String id){this.id=id;}public String getNotepadContent(){return notepadContent;}public void setNotepadContent(String notepadContent){this.notepadContent=notepadContent;}public String getNotepadTime(){return notepadTime;}public void setNotepadTime(String notepadTime){this.notepadTime=notepadTime;}
}

注意包名

(6)编写记事本界面列表适配器
由于记事本界面的记录列表是使用ListView控件来展示的,因此需要创建一个数据适配器NotepadAdapter对ListView控件进行数据适配。步骤如下
选中你所创建的项目的包,右击选择New>Package,创建一个adapter包,在adapter包中创建一个NotepadAdapter类继承自BaseAdapter类,并重写getCount(),getItem(),getItemId(),getView()方法,这些方法用于获取Item总数、对应Item对象、Item对象的Id、对应的Item视图,在NotepadAdapter类中创建一个ViewHolder类,在该类中初始化Item界面中的控件,具体代码如下

package com.example.notepad.adapter;import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;import com.example.notepad.R;
import com.example.notepad.bean.NotepadBean;import java.util.List;public class NotepadAdapter extends BaseAdapter {private LayoutInflater layoutInflater;private List<NotepadBean> list;public NotepadAdapter(Context context, List<NotepadBean>list){this.layoutInflater=LayoutInflater.from(context);this.list=list;}@Overridepublic int getCount(){return list==null?0:list.size();}@Overridepublic Object getItem(int position){return list.get(position);}@Overridepublic long getItemId(int position){return position;}@Overridepublic View getView(int position,View convertView, ViewGroup parent){ViewHolder viewHolder;if(convertView==null){convertView=layoutInflater.inflate(R.layout.notepad_item_layout,null);viewHolder=new ViewHolder(convertView);convertView.setTag(viewHolder);}else{viewHolder=(ViewHolder)convertView.getTag();}NotepadBean noteInfo=(NotepadBean)getItem(position);viewHolder.tvNoteoadContent.setText(noteInfo.getNotepadContent());viewHolder.tvNotepadTime.setText(noteInfo.getNotepadTime());return convertView;}class ViewHolder{TextView tvNoteoadContent;TextView tvNotepadTime;public ViewHolder(View view){tvNoteoadContent=(TextView)view.findViewById(R.id.item_content);tvNotepadTime=(TextView)view.findViewById(R.id.item_time);}}
}

注意包名

(7)创建数据库
在记事本案例中增删改查记录的数据都是通过操作数据库完成的。因此需要创建数据库类SQLiteHelper与数据库的工具类DBUtils,步骤如下


选中你所创建的项目的包,右击选择New>Package,创建一个utils包,在utils包中创建一个DBUtils类,在该类中定义数据库的名称、表名、数据库版本、数据库表中的列名以及获取当前日期等信息

 package com.example.notepad.utils;
import  java.text.SimpleDateFormat;
import java.util.Date;
public class DBUtils {public static final String DATABASE_NAME="Notepad";public static final String DATAVASE_TABLE="Note";public static final int DATEBASE_VERION=1;public static final String NOTEPAD_ID="id";public static final String NOTEPAD_CONTENT="content";public static final String NOTEPAD_TIME="notetime";public static final String getTime(){SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy年MM月dd日HH:mm:ss");Date date=new Date(System.currentTimeMillis());return simpleDateFormat.format(date);}
}

注意包名


选中你所创建的项目的包,右击选择New>Package,创建一个database包,在database包中创建一个SQLiteHelper类继承自SQLiteOpenHelper类

package com.example.notepad.database;import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import com.example.notepad.bean.NotepadBean;
import com.example.notepad.utils.DBUtils;import java.util.ArrayList;
import java.util.List;public class SQLiteHelper extends SQLiteOpenHelper {private SQLiteDatabase sqLiteDatabase;public SQLiteHelper(Context context){super(context,DBUtils.DATABASE_NAME,null,DBUtils.DATEBASE_VERION);sqLiteDatabase=this.getWritableDatabase();}@Overridepublic void onCreate(SQLiteDatabase db){db.execSQL("create table "+DBUtils.DATAVASE_TABLE+"( "+DBUtils.NOTEPAD_ID+" integer primary key autoincrement ,"+DBUtils.NOTEPAD_CONTENT+" text , "+DBUtils.NOTEPAD_TIME+" text ) " );}@Overridepublic  void onUpgrade(SQLiteDatabase db,int oldVersion,int newVersion){}public boolean insertData(String userContent,String userTime){ContentValues contentValues=new ContentValues();contentValues.put(DBUtils.NOTEPAD_CONTENT,userContent);contentValues.put(DBUtils.NOTEPAD_TIME,userTime);return sqLiteDatabase.insert(DBUtils.DATAVASE_TABLE,null,contentValues)>0;}public boolean deteleData(String id){String sql=DBUtils.NOTEPAD_ID+"=?";String[] contentValuesArray=new String[]{String.valueOf(id)};return sqLiteDatabase.delete(DBUtils.DATAVASE_TABLE,sql,contentValuesArray)>0;}public boolean updateData(String id,String content,String userYear){ContentValues contentValues=new ContentValues();contentValues.put(DBUtils.NOTEPAD_CONTENT,content);contentValues.put(DBUtils.NOTEPAD_TIME,userYear);String sql=DBUtils.NOTEPAD_ID+"=?";String[] strings=new String[]{id};return sqLiteDatabase.update(DBUtils.DATAVASE_TABLE,contentValues,sql,strings)>0;}public List<NotepadBean>query(){List<NotepadBean> list=new ArrayList<NotepadBean>();Cursor cursor=sqLiteDatabase.query(DBUtils.DATAVASE_TABLE,null,null,null,null,null,DBUtils.NOTEPAD_ID+" desc");if(cursor!=null){while (cursor.moveToNext()){NotepadBean noteInfo=new NotepadBean();String id=String.valueOf(cursor.getInt(cursor.getColumnIndex(DBUtils.NOTEPAD_ID)));String content=cursor.getString(cursor.getColumnIndex(DBUtils.NOTEPAD_CONTENT));String time=cursor.getString(cursor.getColumnIndex(DBUtils.NOTEPAD_TIME));noteInfo.setId(id);noteInfo.setNotepadContent(content);noteInfo.setNotepadTime(time);list.add(noteInfo);}cursor.close();}return list;}
}

注意包名

(7)实现记事本界面的显示功能
在NotepadActivity中通过创建一个showQueryData()方法查询数据库中存放的记录信息,并将该信息显示到记录列表中,同时在NotepadActivity中还实现了添加按钮的点击事件

package com.example.notepad;import android.content.DialogInterface;
import android.content.Intent;
import android.support.v7.app.AlertDialog;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.Toast;import com.example.notepad.adapter.NotepadAdapter;
import com.example.notepad.bean.NotepadBean;
import com.example.notepad.database.SQLiteHelper;import java.util.List;public class NotepadActivity extends AppCompatActivity {ListView listView;List<NotepadBean> list;SQLiteHelper mSQLiteHelper;NotepadAdapter adapter;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_notepad);listView=(ListView)findViewById(R.id.listview);ImageView add=(ImageView)findViewById(R.id.add);add.setOnClickListener(new View.OnClickListener(){@Overridepublic void onClick(View v){Intent intent=new Intent(NotepadActivity.this,RecordActivity.class);startActivityForResult(intent,1);}});initData();}protected void initData(){mSQLiteHelper=new SQLiteHelper(this);showQueryData();listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> parent, View view, int position, long id) {NotepadBean notepadBean=list.get(position);Intent intent=new Intent(NotepadActivity.this,RecordActivity.class);intent.putExtra("id",notepadBean.getId());intent.putExtra("time",notepadBean.getNotepadTime());intent.putExtra("content",notepadBean.getNotepadContent());NotepadActivity.this.startActivityForResult(intent,1);}});listView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() {@Overridepublic boolean onItemLongClick(AdapterView<?> parent, View view, final int position, long id) {AlertDialog dialog;AlertDialog.Builder builder=new AlertDialog.Builder(NotepadActivity.this).setMessage("是否删除此记录").setPositiveButton("确定", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {NotepadBean notepadBean=list.get(position);if(mSQLiteHelper.deteleData(notepadBean.getId())){list.remove(position);adapter.notifyDataSetChanged();Toast.makeText(NotepadActivity.this,"删除成功",Toast.LENGTH_SHORT).show();}}}).setNegativeButton("取消", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {dialog.dismiss();}});dialog=builder.create();dialog.show();return  true;}});}private void showQueryData(){if(list!=null){list.clear();}list=mSQLiteHelper.query();adapter=new NotepadAdapter(this,list);listView.setAdapter(adapter);}@Overrideprotected  void onActivityResult(int requestCode,int resultCode,Intent data){super.onActivityResult(requestCode,resultCode,data);if(requestCode==1&&resultCode==2){showQueryData();;}}
}

注意包名

(8)搭建添加记录和修改记录界面的布局
当点击记事本界面的添加按钮时,会跳转到添加记录界面,当点击记事本界面列表中的Item时,会跳转到修改记录界面。由于这两个界面上的控件与功能基本相同,因此可以使用同一个Activity和同一个布局文件显示这两个界面,步骤如下

选中你所创建的项目的包,创建一个名为RecordActivity的Activity并将布局文件指名为activity_record,在activity_record.xml布局文件中,放置两个TextView控件,分别用于显示界面标题和记录时间,一个EditText控件用于显示输入框,三个ImageView控件分别用于显示后退键图标、删除图标以及保存按钮图标

1.实现添加记录界面的功能
由于添加记录界面的清除和保存按钮需要实现点击事件,因此将RecordActivity实现View.OnClickListener接口并重写onClick()方法,在该方法中实现将编写的记录添加到数据库中的功能

package com.example.notepad;import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;import com.example.notepad.database.SQLiteHelper;
import com.example.notepad.utils.DBUtils;public class RecordActivity extends AppCompatActivity implements View.OnClickListener {ImageView note_back;TextView note_time;EditText content;ImageView deldete;ImageView note_save;SQLiteHelper mSQLiteHelper;TextView noteName;String id;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_record);note_back = (ImageView) findViewById(R.id.note_back);note_time = (TextView) findViewById(R.id.tv_time);content = (EditText) findViewById(R.id.note_content);deldete = (ImageView) findViewById(R.id.delete);note_save = (ImageView) findViewById(R.id.note_save);noteName = (TextView) findViewById(R.id.note_name);note_back.setOnClickListener(this);deldete.setOnClickListener(this);note_save.setOnClickListener(this);initData();}protected void initData() {mSQLiteHelper = new SQLiteHelper(this);noteName.setText("添加记录");Intent intent = getIntent();if (intent != null) {id = intent.getStringExtra("id");if (id != null) {noteName.setText("修改记录");content.setText(intent.getStringExtra("content"));note_time.setText(intent.getStringExtra("time"));note_time.setVisibility(View.VISIBLE);}}}@Overridepublic void onClick(View v) {switch (v.getId()) {case R.id.note_back:finish();break;case R.id.delete:content.setText("");break;case R.id.note_save:String noteContent = content.getText().toString().trim();if (id != null) {if (noteContent.length() > 0) {if (mSQLiteHelper.updateData(id, noteContent, DBUtils.getTime())) {showToast("修改成功");setResult(2);finish();} else {showToast("修改失败");}} else {showToast("修改内容不能为空");}}else{if(noteContent.length()>0){if(mSQLiteHelper.insertData(noteContent,DBUtils.getTime())){showToast("保存成功");setResult(2);finish();}else{showToast("保存失败");}}else {showToast("修改内容不能为空");}}break;}}public void showToast(String message){Toast.makeText(RecordActivity.this,message,Toast.LENGTH_SHORT).show();}
}

注意包名

2.实现修改记录界面的功能
修改记录界面主要包含查看记录和修改记录的功能

  • 实现查看记录功能
    记事本界面列表的每个Item最多只显示2行记录信息,如果想要查看更多的记录内容,则需要点击Item,进入到修改记录界面进行查看,因此点击Item时,需要将Item对应的记录信息传递到修改记录界面进行显示,即为NotepadActivity的initData()方法,具体代码已在步骤(7)中写出
    在RecordActivity的initData()方法中需要接收记事本界面传递过来的记录数据并将数据显示到界面上,具体代码已在当前步骤中写出
  • 实现修改记录功能
    在RecordActivity中的onClick()方法中,找到保存按钮的点击事件,在该事件中根据判断传递过来的id是否为null来判断处理的是添加记录界面的保存功能还是修改记录界面的保存功能,如果id不为null,则处理修改记录界面的保存功能,具体代码已在当前步骤中写出

(9)删除记事本中的记录
当需要删除记事本列表中的记录时,需要长按列表中的Item,此时会弹出一个对话框提示是否删除Item的对应的记录,因此在NotepadActivity中的initData()方法中写出了删除记录的逻辑代码,具体代码已在步骤(7)中写出

(10)记事本案例已完成

开发一个简单APP的流程及记事本案例(Android Studio)相关推荐

  1. 一个html写的app首页,如何快速开发一个简单好看的APP控制页面

    原标题:如何快速开发一个简单好看的APP控制页面 导读 机智云开源框架为了让开发者快速开发APP,已将用户登录,设备发现,设备配网等功能做成了各个标准模块,仅保留控制页面让开发者自行开发设计,节省了开 ...

  2. 一个简单的审批流程系统设计

    一个简单的审批流程系统设计 1 背景 ​ 最近在做一个企业管理系统的外包,该管理系统主要分为两个端,管理端(web端)和生产端(移动端).管理端的功能有人员管理.项目管理.工作量管理.审批流程管理等, ...

  3. MATLAB Appdesigner开发独立桌面App全流程(二):以实时时间显示为例介绍Timer和StartupFcn的使用以及try catch抛出错误

    1.以实时显示时间为例简单介绍Timer的使用 根据目前所了解到的资料,MATLAB调用多线程较为麻烦,并且类似parfor等语法只适用于大规模运算,而不适合两个独立的.需要并行的任务.这时,我们就需 ...

  4. [译]使用 Rust 开发一个简单的 Web 应用,第 4 部分 —— CLI 选项解析

    原文地址:A Simple Web App in Rust, Part 4 -- CLI Option Parsing 原文作者:Joel's Journal 译文出自:掘金翻译计划 本文永久链接:g ...

  5. php开发mvc教程,php开发一个简单的MVC

    本文通过实例为大家介绍用php开发一个简单mvc的方法,起到势砖引玉的作用,本文比较适合刚接触mvc的朋友. MVC其实就是三个Model,Contraller,View单词的简称. Model,主要 ...

  6. python可视化界面编程 pycharm_pycharm开发一个简单界面和通用mvc模板(操作方法图解)...

    文章首先使用pycharm的 PyQt5 Designer 做一个简单的界面,然后引入所谓的"mvc框架". 一.设计登录界面 下面开始第一个话题,使用pycharm的 PyQt5 ...

  7. 自己怎么开发一个软件app、如何开发一个app系统软件?

    自己怎么开发一个软件app.如何开发一个app系统软件? ​华盛恒辉开发app软件的办法如下: 1.华盛恒辉首先本人明白需求,懂代码,熟习开发流程. 2.华盛恒辉APP开发后期需求理解产品定位. 3. ...

  8. django之十一--开发一个简单的醉得意菜单和人均支付金额查询页面

    一.前言 针对搭伙吃饭的点菜,我们时常会从人数和人均金额去考虑一家餐厅的菜单,所以点菜对于大锅饭的伙伴也成了一门学问和难题,本篇文章巧妙的应用代码思想,将实际问题转化为代码问题,例如如下是一个醉得意的 ...

  9. 开发一个软件的主要流程

    本文重点解决如下问题:开发一个软件的主要流程是什么?了解开发一个软件的主要流程对于编程者而言非常重要,它能够让编程者对如何开发一个软件有个整体的认知.开发一个软件的主要流程包括:1)软件前端界面设计: ...

最新文章

  1. [译]编写优雅的JavaScript代码 - 最佳实践
  2. 计算机地质应用软件,中国地质大学《质软件应用》作业报告.doc
  3. python set使用
  4. 2022-03-16
  5. 补充小知识:文件句柄与文件标识符
  6. LeetCode 436. 寻找右区间(二分查找)
  7. Android Proguard Questions
  8. MySQL 存储过程 if语句
  9. c# mysql 1062_C#中MySQL函数用DATASET 和 MySqlDataAdapter 操作数据库
  10. 【C/C++】变量的内存分配
  11. VALSE学习(十四):自主学习
  12. ssh登录很慢,登录上去后速度正常问题的解决方法
  13. 广和通LTE Cat4模组L716焕新升级,为IoT行业提供经济普适无线应用
  14. zkPorter:Layer-2 的可组合可扩展性
  15. 016画笔工具、铅笔工具、颜色替换工具和混合器画笔工具
  16. CSS实现空心三角指示箭头
  17. 用python画蜡笔小新的步骤_#6.1# 用python画出你的童年回忆
  18. 商业智能,数据仓库,ETL,数仓调度工具informatica介绍手账(三)
  19. linux服务器开发三(网络编程)
  20. 在Oracle中,如何得到真实的执行计划?

热门文章

  1. xp共享文件服务器服务,xp系统共享文件时提示没有启动文件服务器服务怎么办...
  2. Win10无线网络WiFi提示无法连接此网络的原因与解决方案
  3. 云原生钻石课程|第8课:Kubernetes运维管理详解(上)
  4. 新能源汽车引发的电池回收危机
  5. 单元化架构在金融行业的最佳实践
  6. win7计算机打开一直在搜索,在win7电脑中打开文件夹却变成了搜索界面怎么办?...
  7. 墨器杯垫 文创商品设计特优
  8. 这些常见面食的加工机械你见过吗?
  9. python判断手机号码是否正确_Python程序验证输入的电话号码是否正确
  10. 《精彩网址大全——生活资讯文体娱乐卷》内容简介